/******************************************************************************* * Copyright (c) 2000, 2009 IBM Corporation and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation *******************************************************************************/ package org.eclipse.swt.graphics; import org.eclipse.swt.internal.*; import org.eclipse.swt.internal.wpf.*; import org.eclipse.swt.*; /** * <code>TextLayout</code> is a graphic object that represents * styled text. * <p> * Instances of this class provide support for drawing, cursor * navigation, hit testing, text wrapping, alignment, tab expansion * line breaking, etc. These are aspects required for rendering internationalized text. * </p><p> * Application code must explicitly invoke the <code>TextLayout#dispose()</code> * method to release the operating system resources managed by each instance * when those instances are no longer required. * </p> * * @see <a href="http://www.eclipse.org/swt/snippets/#textlayout">TextLayout, TextStyle snippets</a> * @see <a href="http://www.eclipse.org/swt/examples.php">SWT Example: CustomControlExample, StyledText tab</a> * @see <a href="http://www.eclipse.org/swt/">Sample code and further information</a> * * @since 3.0 */ public final class TextLayout extends Resource { Font font; String text, segmentsText; int lineSpacing; int ascent, descent; int alignment; int wrapWidth; int orientation; int indent; int wrapIndent; boolean justify; int[] tabs; int[] segments; char[] segmentsChars; StyleItem[] styles; int string, defaultTextProperties; int[] runs; int[] lines; static final RGB LINK_FOREGROUND = new RGB (0, 51, 153); static final char LTR_MARK = '\u200E', RTL_MARK = '\u200F'; static final int TAB_COUNT = 32; class StyleItem { TextStyle style; int start, length; int textProperties; void free() { if (textProperties != 0) OS.GCHandle_Free(textProperties); textProperties = 0; } public String toString () { return "StyleItem {" + start + ", " + style + "}"; } } /** * Constructs a new instance of this class on the given device. * <p> * You must dispose the text layout when it is no longer required. * </p> * * @param device the device on which to allocate the text layout * * @exception IllegalArgumentException <ul> * <li>ERROR_NULL_ARGUMENT - if device is null and there is no current device</li> * </ul> * * @see #dispose() */ public TextLayout (Device device) { super(device); wrapWidth = ascent = descent = -1; lineSpacing = 0; orientation = SWT.LEFT_TO_RIGHT; styles = new StyleItem[2]; styles[0] = new StyleItem(); styles[1] = new StyleItem(); text = ""; //$NON-NLS-1$ init(); } void checkLayout () { if (isDisposed()) SWT.error(SWT.ERROR_GRAPHIC_DISPOSED); } /* * Compute the runs: itemize, shape, place, and reorder the runs. * Break paragraphs into lines, wraps the text, and initialize caches. */ void computeRuns () { if (lines != null) return; int jniRef = OS.NewGlobalRef(this); int textSource = OS.gcnew_SWTTextSource(jniRef); int formatter = OS.TextFormatter_Create(); Font font = this.font != null ? this.font : device.systemFont; segmentsText = getSegmentsText(); int length = segmentsText.length(); char [] buffer = new char [length]; segmentsText.getChars (0, length, buffer, 0); string = OS.gcnew_String(buffer, 0 ,length); int culture = OS.CultureInfo_CurrentUICulture(); defaultTextProperties = OS.gcnew_SWTTextRunProperties(font.handle, font.size, font.size, 0, 0, 0, OS.BaselineAlignment_Baseline, culture); for (int i = 0; i < styles.length; i++) { StyleItem run = styles[i]; TextStyle style = run.style; if (style != null) { Font styleFont = style.font != null ? style.font : font; int fg = 0; if (style.foreground != null) { fg = OS.gcnew_SolidColorBrush(style.foreground.handle); } else { if (style.underline && style.underlineStyle == SWT.UNDERLINE_LINK) { int color = OS.Color_FromArgb((byte)0xFF, (byte)LINK_FOREGROUND.red, (byte)LINK_FOREGROUND.green, (byte)LINK_FOREGROUND.blue); fg = OS.gcnew_SolidColorBrush(color); OS.GCHandle_Free(color); } } int bg = 0; if (style.background != null) { bg = OS.gcnew_SolidColorBrush(style.background.handle); } int decorations = 0; if (style.strikeout || style.underline) { decorations = OS.gcnew_TextDecorationCollection(2); if (style.strikeout) { int pen = 0; if (style.strikeoutColor != null) { int color = style.strikeoutColor.handle; int brush = OS.gcnew_SolidColorBrush(color); pen = OS.gcnew_Pen(brush, 1); OS.GCHandle_Free(brush); } int strikeout = OS.gcnew_TextDecoration(OS.TextDecorationLocation_Strikethrough, pen, 0, OS.TextDecorationUnit_FontRecommended, OS.TextDecorationUnit_FontRecommended); OS.TextDecorationCollection_Add(decorations, strikeout); OS.GCHandle_Free(strikeout); if (pen != 0) OS.GCHandle_Free(pen); } if (style.underline) { int brush; if (style.underlineColor != null) { Color color = style.underlineColor; brush = OS.gcnew_SolidColorBrush(color.handle); } else { if (fg != 0) { brush = fg; } else { brush = OS.Brushes_Black(); } } int pen = OS.gcnew_Pen(brush, 1f); if (brush != fg) OS.GCHandle_Free(brush); int underline; switch (style.underlineStyle) { case SWT.UNDERLINE_SQUIGGLE: //TODO implement case SWT.UNDERLINE_ERROR: int dashStyle = OS.DashStyles_Dash(); OS.Pen_DashStyle(pen, dashStyle); underline = OS.gcnew_TextDecoration(OS.TextDecorationLocation_Underline, pen, 0, OS.TextDecorationUnit_FontRecommended, OS.TextDecorationUnit_FontRecommended); OS.TextDecorationCollection_Add(decorations, underline); OS.GCHandle_Free(underline); OS.GCHandle_Free(dashStyle); break; case SWT.UNDERLINE_DOUBLE: underline = OS.gcnew_TextDecoration(OS.TextDecorationLocation_Underline, pen, 1, OS.TextDecorationUnit_FontRecommended, OS.TextDecorationUnit_FontRecommended); OS.TextDecorationCollection_Add(decorations, underline); OS.GCHandle_Free(underline); //FALLTHROU case SWT.UNDERLINE_LINK: case SWT.UNDERLINE_SINGLE: underline = OS.gcnew_TextDecoration(OS.TextDecorationLocation_Underline, pen, 0, OS.TextDecorationUnit_FontRecommended, OS.TextDecorationUnit_FontRecommended); OS.TextDecorationCollection_Add(decorations, underline); OS.GCHandle_Free(underline); break; } if (pen != 0) OS.GCHandle_Free(pen); } } run.textProperties = OS.gcnew_SWTTextRunProperties(styleFont.handle, styleFont.size, styleFont.size, decorations, fg, bg, OS.BaselineAlignment_Baseline, culture); if (fg != 0) OS.GCHandle_Free(fg); if (bg != 0) OS.GCHandle_Free(bg); if (decorations != 0) OS.GCHandle_Free(decorations); } } int textAlignment = OS.TextAlignment_Left; if (justify) { textAlignment = OS.TextAlignment_Justify; } else { switch (alignment) { case SWT.CENTER: textAlignment = OS.TextAlignment_Center; break; case SWT.RIGHT: textAlignment = OS.TextAlignment_Right; break; } } int flowDirection = (orientation & SWT.RIGHT_TO_LEFT) != 0 ? OS.FlowDirection_RightToLeft : OS.FlowDirection_LeftToRight; int textWrapping = wrapWidth != -1 ? OS.TextWrapping_Wrap : OS.TextWrapping_NoWrap; int tabCollection = 0; if (tabs != null) { int position = 0; int tabLength = Math.max(tabs.length, TAB_COUNT), i; tabCollection = OS.gcnew_TextTabPropertiesCollection(tabLength); for (i = 0; i < tabs.length; i++) { position = tabs[i]; int tab = OS.gcnew_TextTabProperties(OS.TextTabAlignment_Left, position, 0, 0); OS.TextTabPropertiesCollection_Add(tabCollection, tab); OS.GCHandle_Free(tab); } int width = tabs[tabs.length - 1]; if (tabs.length > 1) width -= tabs[tabs.length - 2]; if (width > 0) { for (; i < length; i++) { position += width; int tab = OS.gcnew_TextTabProperties(OS.TextTabAlignment_Left, position, 0, 0); OS.TextTabPropertiesCollection_Add(tabCollection, tab); OS.GCHandle_Free(tab); } } } int paragraphProperties = OS.gcnew_SWTTextParagraphProperties(flowDirection, textAlignment, false, defaultTextProperties, textWrapping, 0, wrapIndent, tabCollection); int firstParagraphProperties = OS.gcnew_SWTTextParagraphProperties(flowDirection, textAlignment, true, defaultTextProperties, textWrapping, 0, indent, tabCollection); int offset = 0; int index = 0; lines = new int[4]; int lineBreak = 0; while (offset < length || offset == 0) { char ch; boolean firstLine = offset == 0 || (ch = segmentsText.charAt(offset - 1)) == '\r' || ch == '\n'; int paragraphProps = firstLine ? firstParagraphProperties : paragraphProperties; int textLine = OS.TextFormatter_FormatLine(formatter, textSource, offset, wrapWidth != -1 ? wrapWidth : 0, paragraphProps, lineBreak); offset += OS.TextLine_Length(textLine); lineBreak = OS.TextLine_GetTextLineBreak(textLine); if (index == lines.length) { int[] tmpLines = new int[index + 4]; System.arraycopy(lines, 0, tmpLines, 0, index); lines = tmpLines; } lines[index++] = textLine; } if (index != lines.length) { int[] tmpLines = new int[index]; System.arraycopy(lines, 0, tmpLines, 0, index); lines = tmpLines; } if (tabCollection != 0) OS.GCHandle_Free(tabCollection); OS.GCHandle_Free(paragraphProperties); OS.GCHandle_Free(firstParagraphProperties); OS.GCHandle_Free(culture); OS.GCHandle_Free(formatter); OS.GCHandle_Free(textSource); OS.DeleteGlobalRef(jniRef); } void destroy() { freeRuns(); font = null; text = null; segmentsText = null; tabs = null; styles = null; segments = null; segmentsChars = null; // lineOffset = null; // lineY = null; // lineWidth = null; } /** * Draws the receiver's text using the specified GC at the specified * point. * * @param gc the GC to draw * @param x the x coordinate of the top left corner of the rectangular area where the text is to be drawn * @param y the y coordinate of the top left corner of the rectangular area where the text is to be drawn * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * @exception IllegalArgumentException <ul> * <li>ERROR_NULL_ARGUMENT - if the gc is null</li> * </ul> */ public void draw (GC gc, int x, int y) { draw(gc, x, y, -1, -1, null, null); } /** * Draws the receiver's text using the specified GC at the specified * point. * * @param gc the GC to draw * @param x the x coordinate of the top left corner of the rectangular area where the text is to be drawn * @param y the y coordinate of the top left corner of the rectangular area where the text is to be drawn * @param selectionStart the offset where the selections starts, or -1 indicating no selection * @param selectionEnd the offset where the selections ends, or -1 indicating no selection * @param selectionForeground selection foreground, or NULL to use the system default color * @param selectionBackground selection background, or NULL to use the system default color * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * @exception IllegalArgumentException <ul> * <li>ERROR_NULL_ARGUMENT - if the gc is null</li> * </ul> */ public void draw (GC gc, int x, int y, int selectionStart, int selectionEnd, Color selectionForeground, Color selectionBackground) { draw(gc, x, y, selectionStart, selectionEnd, selectionForeground, selectionBackground, 0); } /** * Draws the receiver's text using the specified GC at the specified * point. * <p> * The parameter <code>flags</code> can include one of <code>SWT.DELIMITER_SELECTION</code> * or <code>SWT.FULL_SELECTION</code> to specify the selection behavior on all lines except * for the last line, and can also include <code>SWT.LAST_LINE_SELECTION</code> to extend * the specified selection behavior to the last line. * </p> * @param gc the GC to draw * @param x the x coordinate of the top left corner of the rectangular area where the text is to be drawn * @param y the y coordinate of the top left corner of the rectangular area where the text is to be drawn * @param selectionStart the offset where the selections starts, or -1 indicating no selection * @param selectionEnd the offset where the selections ends, or -1 indicating no selection * @param selectionForeground selection foreground, or NULL to use the system default color * @param selectionBackground selection background, or NULL to use the system default color * @param flags drawing options * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * @exception IllegalArgumentException <ul> * <li>ERROR_NULL_ARGUMENT - if the gc is null</li> * </ul> * * @since 3.3 */ public void draw (GC gc, int x, int y, int selectionStart, int selectionEnd, Color selectionForeground, Color selectionBackground, int flags) { checkLayout(); computeRuns(); if (gc == null) SWT.error(SWT.ERROR_NULL_ARGUMENT); if (gc.isDisposed()) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (selectionForeground != null && selectionForeground.isDisposed()) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (selectionBackground != null && selectionBackground.isDisposed()) SWT.error(SWT.ERROR_INVALID_ARGUMENT); int length = text.length(); if (length == 0 && flags == 0) return; gc.checkGC(GC.FOREGROUND); int fg = OS.Pen_Brush(gc.data.pen); OS.SWTTextRunProperties_ForegroundBrush(defaultTextProperties, fg); for (int i = 0; i < styles.length; i++) { StyleItem run = styles[i]; if (run.textProperties == 0) continue; if (run.style != null && run.style.foreground != null) continue; if (run.style != null && run.style.underline && run.style.underlineStyle == SWT.UNDERLINE_LINK) continue; OS.SWTTextRunProperties_ForegroundBrush(run.textProperties, fg); } int drawingContext = gc.handle; boolean hasSelection = selectionStart <= selectionEnd && selectionStart != -1 && selectionEnd != -1; int selBrush = 0, selGeometry = 0, geometries = 0; if (hasSelection || (flags & SWT.LAST_LINE_SELECTION) != 0) { selectionStart = Math.min(Math.max(0, selectionStart), length - 1); selectionEnd = Math.min(Math.max(0, selectionEnd), length - 1); selectionStart = translateOffset(selectionStart); selectionEnd = translateOffset(selectionEnd); if (selectionBackground != null) { selBrush = OS.gcnew_SolidColorBrush(selectionBackground.handle); } else { selBrush = OS.Brushes_LightSkyBlue(); } selGeometry = OS.gcnew_GeometryGroup(); geometries = OS.GeometryGroup_Children(selGeometry); } int lineStart = 0, lineEnd = 0; double drawY = y; for (int i = 0; i < lines.length; i++) { int line = lines[i]; if (line == 0) break; lineStart = lineEnd; lineEnd = lineStart + OS.TextLine_Length(line); double nextDrawY, selY = drawY; int lineHeight = (int)OS.TextLine_Height(line); if (ascent != -1 && descent != -1) { lineHeight = Math.max(lineHeight, ascent + descent); nextDrawY = drawY + lineHeight + lineSpacing; int baseline = (int)OS.TextLine_Baseline(line); if (ascent > baseline) drawY += ascent - baseline; } else { nextDrawY = drawY + lineHeight + lineSpacing; } //draw line text int point = OS.gcnew_Point(x, drawY); OS.TextLine_Draw(line, drawingContext, point, 0); OS.GCHandle_Free(point); //draw line selection boolean fullSelection = selectionStart <= lineStart && selectionEnd >= lineEnd; boolean partialSelection = !(selectionStart > lineEnd || lineStart > selectionEnd); if (flags != 0 && (hasSelection || (flags & SWT.LAST_LINE_SELECTION) != 0)) { boolean extent = false; if (i == lines.length - 1 && (flags & SWT.LAST_LINE_SELECTION) != 0) { extent = true; } else { int breakLength = OS.TextLine_NewlineLength(line); if (breakLength != 0) { if (selectionStart <= lineEnd && lineEnd <= selectionEnd) extent = true; } else { if (selectionStart <= lineEnd && lineEnd < selectionEnd && (flags & SWT.FULL_SELECTION) != 0) { extent = true; } } } if (extent) { int extentWidth = (flags & SWT.FULL_SELECTION) != 0 ? 0x7ffffff : lineHeight / 3; int textRect = OS.gcnew_Rect(OS.TextLine_WidthIncludingTrailingWhitespace(line) + x, selY, extentWidth, lineHeight); int geometry = OS.gcnew_RectangleGeometry(textRect); OS.GeometryCollection_Add(geometries, geometry); OS.GCHandle_Free(geometry); OS.GCHandle_Free(textRect); } } if (hasSelection && (fullSelection || partialSelection)) { int selLineStart = Math.max (lineStart, selectionStart); int selLineEnd = Math.min (lineEnd, selectionEnd); int rects = OS.TextLine_GetTextBounds(line, selLineStart, selLineEnd - selLineStart + 1); if (rects != 0) { int enumerator = OS.TextBoundsCollection_GetEnumerator(rects); while (OS.IEnumerator_MoveNext(enumerator)) { int bounds = OS.TextBoundsCollection_Current(enumerator); int textRect = OS.TextBounds_Rectangle(bounds); OS.Rect_X(textRect, OS.Rect_X(textRect) + x); OS.Rect_Y(textRect, selY); OS.Rect_Height(textRect, lineHeight); int geometry = OS.gcnew_RectangleGeometry(textRect); OS.GeometryCollection_Add(geometries, geometry); OS.GCHandle_Free(geometry); OS.GCHandle_Free(textRect); OS.GCHandle_Free(bounds); } OS.GCHandle_Free(enumerator); } OS.GCHandle_Free(rects); } drawY = nextDrawY; } for (int i = 0; i < styles.length - 1; i++) { StyleItem run = styles[i]; TextStyle style = run.style; if (style == null) continue; if (style.borderStyle != SWT.NONE && (i + 1 >= styles.length || !style.isAdherentBorder(styles[i + 1].style))) { int start = run.start; int end = styles[i + 1].start - 1; for (int j = i; j > 0 && style.isAdherentBorder(styles[j - 1].style); j--) { start = styles[j - 1].start; } Color color = style.borderColor; if (color == null) color = style.foreground; if (color == null) color = gc.getForeground(); int brush = OS.gcnew_SolidColorBrush(color.handle); int pen = OS.gcnew_Pen(brush, 1); OS.GCHandle_Free(brush); int dashStyle = 0; switch (style.borderStyle) { case SWT.BORDER_SOLID: dashStyle = OS.DashStyles_Solid(); break; case SWT.BORDER_DOT: dashStyle = OS.DashStyles_Dot(); break; case SWT.BORDER_DASH: dashStyle = OS.DashStyles_Dash(); break; } OS.Pen_DashStyle(pen, dashStyle); if (dashStyle != 0) OS.GCHandle_Free(dashStyle); int lineY = y; lineStart = lineEnd = 0; for (int j = 0; j < lines.length; j++) { int lineLength = OS.TextLine_Length(lines[j]); lineStart = lineEnd; lineEnd = lineStart + lineLength; if (start < lineEnd) { if (end < lineStart) break; int rangeStart = Math.max(start, lineStart); int rangLength = Math.min(end, lineEnd) - rangeStart + 1; int rects = OS.TextLine_GetTextBounds(lines[j], rangeStart, rangLength); if (rects != 0) { int enumerator = OS.TextBoundsCollection_GetEnumerator(rects); while (OS.IEnumerator_MoveNext(enumerator)) { int bounds = OS.TextBoundsCollection_Current(enumerator); int textRect = OS.TextBounds_Rectangle(bounds); OS.Rect_Y(textRect, OS.Rect_Y(textRect) + lineY); OS.Rect_X(textRect, OS.Rect_X(textRect) + x); OS.Rect_Width(textRect, OS.Rect_Width(textRect) - 1); OS.Rect_Height(textRect, OS.Rect_Height(textRect) - 1); OS.DrawingContext_DrawRectangle(drawingContext, 0, pen, textRect); OS.GCHandle_Free(textRect); OS.GCHandle_Free(bounds); } OS.GCHandle_Free(enumerator); } OS.GCHandle_Free(rects); } int lineHeight = (int)OS.TextLine_Height(lines[j]); if (ascent != -1 && descent != -1) lineHeight = Math.max(lineHeight, ascent + descent); lineY += lineHeight + lineSpacing; } OS.GCHandle_Free(pen); } } if (selGeometry != 0) { OS.DrawingContext_PushOpacity(drawingContext, 0.4); OS.DrawingContext_DrawGeometry(drawingContext, selBrush, 0, selGeometry); OS.DrawingContext_Pop(drawingContext); OS.GCHandle_Free(geometries); OS.GCHandle_Free(selGeometry); } if (selBrush != 0) OS.GCHandle_Free(selBrush); OS.GCHandle_Free(fg); } void freeRuns () { if (lines == null) return; for (int i = 0; i < lines.length; i++) { if (lines[i] != 0) { OS.GCHandle_Free(lines[i]); } } lines = null; if (runs != null) { for (int i = 0; i < runs.length; i++) { if (runs[i] == 0) break; OS.GCHandle_Free(runs[i]); } runs = null; } for (int i = 0; i < styles.length; i++) { styles[i].free(); } if (defaultTextProperties != 0) OS.GCHandle_Free(defaultTextProperties); if (string != 0) OS.GCHandle_Free(string); defaultTextProperties = string = 0; segmentsText = null; } /** * Returns the receiver's horizontal text alignment, which will be one * of <code>SWT.LEFT</code>, <code>SWT.CENTER</code> or * <code>SWT.RIGHT</code>. * * @return the alignment used to positioned text horizontally * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getAlignment () { checkLayout(); return alignment; } /** * Returns the ascent of the receiver. * * @return the ascent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getDescent() * @see #setDescent(int) * @see #setAscent(int) * @see #getLineMetrics(int) */ public int getAscent () { checkLayout(); return ascent; } /** * Returns the bounds of the receiver. The width returned is either the * width of the longest line or the width set using {@link TextLayout#setWidth(int)}. * To obtain the text bounds of a line use {@link TextLayout#getLineBounds(int)}. * * @return the bounds of the receiver * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setWidth(int) * @see #getLineBounds(int) */ public Rectangle getBounds () { checkLayout(); computeRuns(); double width = 0; double height = 0; for (int line=0; line<lines.length; line++) { if (wrapWidth == -1) width = Math.max(width, OS.TextLine_WidthIncludingTrailingWhitespace(lines[line])); int lineHeight = (int)OS.TextLine_Height(lines[line]); if (ascent != -1 && descent != -1) lineHeight = Math.max(lineHeight, ascent + descent); height += lineHeight + lineSpacing; } if (wrapWidth != -1) width = wrapWidth; return new Rectangle (0, 0, (int)width, (int)height); } /** * Returns the bounds for the specified range of characters. The * bounds is the smallest rectangle that encompasses all characters * in the range. The start and end offsets are inclusive and will be * clamped if out of range. * * @param start the start offset * @param end the end offset * @return the bounds of the character range * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public Rectangle getBounds (int start, int end) { checkLayout(); computeRuns(); int length = text.length(); if (length == 0) return new Rectangle(0, 0, 0, 0); if (start > end) return new Rectangle(0, 0, 0, 0); start = Math.min(Math.max(0, start), length - 1); end = Math.min(Math.max(0, end), length - 1); start = translateOffset(start); end = translateOffset(end); int lineStart = 0, lineEnd = 0, lineY = 0; int rect = 0; for (int i = 0; i < lines.length; i++) { int lineLength = OS.TextLine_Length(lines[i]); lineStart = lineEnd; lineEnd = lineStart + lineLength; if (start < lineEnd) { if (end < lineStart) break; int rangeStart = Math.max(start, lineStart); int rangLength = Math.min(end, lineEnd) - rangeStart + 1; int rects = OS.TextLine_GetTextBounds(lines[i], rangeStart, rangLength); if (rects != 0) { int enumerator = OS.TextBoundsCollection_GetEnumerator(rects); while (OS.IEnumerator_MoveNext(enumerator)) { int bounds = OS.TextBoundsCollection_Current(enumerator); int textRect = OS.TextBounds_Rectangle(bounds); OS.Rect_Y(textRect, OS.Rect_Y(textRect) + lineY); if (rect != 0) { OS.Rect_Union(rect, textRect); OS.GCHandle_Free(textRect); } else { rect = textRect; } OS.GCHandle_Free(bounds); } OS.GCHandle_Free(enumerator); } OS.GCHandle_Free(rects); } int lineHeight = (int)OS.TextLine_Height(lines[i]); if (ascent != -1 && descent != -1) lineHeight = Math.max(lineHeight, ascent + descent); lineY += lineHeight + lineSpacing; } if (rect == 0) return new Rectangle(0, 0, 0, 0); Rectangle result = new Rectangle((int)OS.Rect_X(rect), (int)OS.Rect_Y(rect), (int)OS.Rect_Width(rect), (int)OS.Rect_Height(rect)); OS.GCHandle_Free(rect); return result; } /** * Returns the descent of the receiver. * * @return the descent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getAscent() * @see #setAscent(int) * @see #setDescent(int) * @see #getLineMetrics(int) */ public int getDescent () { checkLayout(); return descent; } /** * Returns the default font currently being used by the receiver * to draw and measure text. * * @return the receiver's font * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public Font getFont () { checkLayout(); return font; } /** * Returns the receiver's indent. * * @return the receiver's indent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @since 3.2 */ public int getIndent () { checkLayout(); return indent; } /** * Returns the receiver's justification. * * @return the receiver's justification * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @since 3.2 */ public boolean getJustify () { checkLayout(); return justify; } /** * Returns the embedding level for the specified character offset. The * embedding level is usually used to determine the directionality of a * character in bidirectional text. * * @param offset the character offset * @return the embedding level * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the character offset is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> */ public int getLevel (int offset) { checkLayout(); computeRuns(); int length = text.length(); if (!(0 <= offset && offset <= length)) SWT.error(SWT.ERROR_INVALID_RANGE); offset = translateOffset(offset); int level = (orientation & SWT.RIGHT_TO_LEFT) != 0 ? 1 : 0; for (int i = 0; i < lines.length; i++) { int lineLength = OS.TextLine_Length(lines[i]); if (lineLength > offset) { int runs = OS.TextLine_GetIndexedGlyphRuns (lines[i]); int enumerator = OS.IndexedGlyphRunCollection_GetEnumerator(runs); while (OS.IEnumerator_MoveNext(enumerator)) { int indexedGlyphRun = OS.IndexedGlyphRunCollection_Current(enumerator); int rangeStart = OS.IndexedGlyphRun_TextSourceCharacterIndex(indexedGlyphRun); int rangeEnd = rangeStart + OS.IndexedGlyphRun_TextSourceLength(indexedGlyphRun); int glyphRun = OS.IndexedGlyphRun_GlyphRun(indexedGlyphRun); int bidiLevel = OS.GlyphRun_BidiLevel(glyphRun); OS.GCHandle_Free(glyphRun); OS.GCHandle_Free(indexedGlyphRun); if (rangeStart <= offset && offset < rangeEnd) { level = bidiLevel; break; } } OS.GCHandle_Free(enumerator); OS.GCHandle_Free(runs); break; } } return level; } /** * Returns the bounds of the line for the specified line index. * * @param lineIndex the line index * @return the line bounds * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the line index is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public Rectangle getLineBounds(int lineIndex) { checkLayout(); computeRuns(); if (!(0 <= lineIndex && lineIndex < runs.length)) SWT.error(SWT.ERROR_INVALID_RANGE); int offset = 0; double y = 0; for (int i=0; i<lineIndex; i++) { offset += OS.TextLine_Length(lines[i]); int lineHeight = (int)OS.TextLine_Height(lines[i]); if (ascent != -1 && descent != -1) lineHeight = Math.max(lineHeight, ascent + descent); y += lineHeight + lineSpacing; } int line = lines[lineIndex]; double x = OS.TextLine_Start(line); double width = OS.TextLine_WidthIncludingTrailingWhitespace(line); double height = OS.TextLine_Height(line); if (ascent != -1 && descent != -1) height = Math.max(height, ascent + descent); char ch; boolean firstLine = offset == 0 || (ch = segmentsText.charAt(offset - 1)) == '\r' || ch == '\n'; if (firstLine) { x += indent; width -= indent; } else { x += wrapIndent; width -= wrapIndent; } return new Rectangle ((int)x, (int)y, (int)width, (int)height); } /** * Returns the receiver's line count. This includes lines caused * by wrapping. * * @return the line count * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getLineCount () { checkLayout(); computeRuns(); return lines.length; } /** * Returns the index of the line that contains the specified * character offset. * * @param offset the character offset * @return the line index * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the character offset is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getLineIndex (int offset) { checkLayout(); computeRuns(); int length = text.length(); if (!(0 <= offset && offset <= length)) SWT.error(SWT.ERROR_INVALID_RANGE); offset = translateOffset(offset); int start = 0; for (int line=0; line<lines.length; line++) { int lineLength = OS.TextLine_Length(lines[line]); if (start + lineLength > offset) return line; start += lineLength; } return lines.length - 1; } /** * Returns the font metrics for the specified line index. * * @param lineIndex the line index * @return the font metrics * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the line index is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public FontMetrics getLineMetrics (int lineIndex) { checkLayout(); computeRuns(); if (!(0 <= lineIndex && lineIndex < runs.length)) SWT.error(SWT.ERROR_INVALID_RANGE); int length = text.length(); double baseline, height; if (length == 0) { Font font = this.font != null ? this.font : device.systemFont; int str = OS.gcnew_String (new char []{' ', '\0'}); int culture = OS.CultureInfo_CurrentUICulture(); int direction = (orientation & SWT.RIGHT_TO_LEFT) != 0 ? OS.FlowDirection_RightToLeft : OS.FlowDirection_LeftToRight; int brush = OS.Brushes_White(); int text = OS.gcnew_FormattedText(str, culture, direction, font.handle, font.size, brush); height = OS.FormattedText_Height(text); baseline = OS.FormattedText_Baseline(text); OS.GCHandle_Free(text); OS.GCHandle_Free(str); OS.GCHandle_Free(brush); OS.GCHandle_Free(culture); } else { baseline = OS.TextLine_Baseline(lines[lineIndex]); height = OS.TextLine_Height(lines[lineIndex]); if (ascent != -1 && descent != -1) { baseline = Math.max(baseline, ascent); height = Math.max(height, ascent + descent); } } return FontMetrics.wpf_new((int)baseline, (int)height - (int)baseline, 0, 0, (int)height); } /** * Returns the line offsets. Each value in the array is the * offset for the first character in a line except for the last * value, which contains the length of the text. * * @return the line offsets * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int[] getLineOffsets () { checkLayout(); computeRuns(); int start = 0; int[] offsets = new int[lines.length+1]; for (int i = 0; i < lines.length; i++) { start += OS.TextLine_Length(lines[i]); offsets[i+1] = untranslateOffset(start); } return offsets; } /** * Returns the location for the specified character offset. The * <code>trailing</code> argument indicates whether the offset * corresponds to the leading or trailing edge of the cluster. * * @param offset the character offset * @param trailing the trailing flag * @return the location of the character offset * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getOffset(Point, int[]) * @see #getOffset(int, int, int[]) */ public Point getLocation (int offset, boolean trailing) { checkLayout(); computeRuns(); int length = text.length(); if (!(0 <= offset && offset <= length)) SWT.error(SWT.ERROR_INVALID_RANGE); offset = translateOffset(offset); double y = 0; int start = 0, line; for (line=0; line<lines.length; line++) { int lineLength = OS.TextLine_Length(lines[line]); if (start + lineLength > offset) break; start += lineLength; int lineHeight = (int)OS.TextLine_Height(lines[line]); if (ascent != -1 && descent != -1) lineHeight = Math.max(lineHeight, ascent + descent); y += lineHeight + lineSpacing; } int characterHit = OS.gcnew_CharacterHit(offset, trailing ? 1 : 0); double x = OS.TextLine_GetDistanceFromCharacterHit(lines[line], characterHit); OS.GCHandle_Free(characterHit); return new Point((int)x, (int)y); } /** * Returns the next offset for the specified offset and movement * type. The movement is one of <code>SWT.MOVEMENT_CHAR</code>, * <code>SWT.MOVEMENT_CLUSTER</code>, <code>SWT.MOVEMENT_WORD</code>, * <code>SWT.MOVEMENT_WORD_END</code> or <code>SWT.MOVEMENT_WORD_START</code>. * * @param offset the start offset * @param movement the movement type * @return the next offset * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the offset is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getPreviousOffset(int, int) */ public int getNextOffset (int offset, int movement) { checkLayout(); return _getOffset (offset, movement, true); } int _getOffset(int offset, int movement, boolean forward) { computeRuns(); int length = text.length(); if (!(0 <= offset && offset <= length)) SWT.error(SWT.ERROR_INVALID_RANGE); if (forward && offset == length) return length; if (!forward && offset == 0) return 0; int step = forward ? 1 : -1; if ((movement & SWT.MOVEMENT_CHAR) != 0) return offset + step; offset = translateOffset(offset); int lineStart = 0, lineIndex; for (lineIndex=0; lineIndex<lines.length; lineIndex++) { int lineLength = OS.TextLine_Length(lines[lineIndex]); if (lineStart + lineLength > offset) break; lineStart += lineLength; } int line = lines[lineIndex]; int lineLength = OS.TextLine_Length(line); int lineBreak = OS.TextLine_NewlineLength (line); while (lineStart <= offset && offset <= lineStart + lineLength) { int resultCharHit; int characterHit = OS.gcnew_CharacterHit(offset, 0); if (forward) { resultCharHit = OS.TextLine_GetNextCaretCharacterHit(line, characterHit); } else { resultCharHit = OS.TextLine_GetPreviousCaretCharacterHit(line, characterHit); } int newOffset = OS.CharacterHit_FirstCharacterIndex(resultCharHit); int trailing = OS.CharacterHit_TrailingLength(resultCharHit); OS.GCHandle_Free(resultCharHit); OS.GCHandle_Free(characterHit); if (forward) { if (newOffset + trailing >= lineStart + lineLength - lineBreak) { int lineEnd = lineStart + lineLength; if (trailing != 0) lineEnd -= lineBreak; return untranslateOffset(Math.min(length, lineEnd)); } } else { if (newOffset + trailing == lineStart) { if (lineIndex == 0) return 0; int lineEnd = 0; if (newOffset + trailing == offset) lineEnd = OS.TextLine_NewlineLength(lines[lineIndex - 1]); return untranslateOffset(Math.max(0, newOffset + trailing - lineEnd)); } } offset = newOffset + trailing; switch (movement) { case SWT.MOVEMENT_CLUSTER: return untranslateOffset(offset); case SWT.MOVEMENT_WORD: case SWT.MOVEMENT_WORD_START: { if (offset > 0) { boolean letterOrDigit = Compatibility.isLetterOrDigit(segmentsText.charAt(offset)); boolean previousLetterOrDigit = Compatibility.isLetterOrDigit(segmentsText.charAt(offset - 1)); if (letterOrDigit != previousLetterOrDigit || !letterOrDigit) { if (!Compatibility.isWhitespace(segmentsText.charAt(offset))) { return untranslateOffset(offset); } } } break; } case SWT.MOVEMENT_WORD_END: { if (offset > 0) { boolean isLetterOrDigit = Compatibility.isLetterOrDigit(segmentsText.charAt(offset)); boolean previousLetterOrDigit = Compatibility.isLetterOrDigit(segmentsText.charAt(offset - 1)); if (!isLetterOrDigit && previousLetterOrDigit) { return untranslateOffset(offset); } } break; } } } return forward ? length : 0; } /** * Returns the character offset for the specified point. * For a typical character, the trailing argument will be filled in to * indicate whether the point is closer to the leading edge (0) or * the trailing edge (1). When the point is over a cluster composed * of multiple characters, the trailing argument will be filled with the * position of the character in the cluster that is closest to * the point. * * @param point the point * @param trailing the trailing buffer * @return the character offset * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the trailing length is less than <code>1</code></li> * <li>ERROR_NULL_ARGUMENT - if the point is null</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getLocation(int, boolean) */ public int getOffset (Point point, int[] trailing) { checkLayout(); if (point == null) SWT.error (SWT.ERROR_NULL_ARGUMENT); return getOffset (point.x, point.y, trailing) ; } /** * Returns the character offset for the specified point. * For a typical character, the trailing argument will be filled in to * indicate whether the point is closer to the leading edge (0) or * the trailing edge (1). When the point is over a cluster composed * of multiple characters, the trailing argument will be filled with the * position of the character in the cluster that is closest to * the point. * * @param x the x coordinate of the point * @param y the y coordinate of the point * @param trailing the trailing buffer * @return the character offset * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the trailing length is less than <code>1</code></li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getLocation(int, boolean) */ public int getOffset (int x, int y, int[] trailing) { checkLayout(); computeRuns(); if (trailing != null && trailing.length < 1) SWT.error(SWT.ERROR_INVALID_ARGUMENT); double lineY = 0; int line; for (line=0; line<lines.length; line++) { double lineHeight = OS.TextLine_Length(lines[line]); if (lineY + lineHeight > y) break; lineY += lineHeight; } if (line >= lines.length) line = lines.length - 1; int characterHit = OS.TextLine_GetCharacterHitFromDistance(lines[line], x); int offset = OS.CharacterHit_FirstCharacterIndex(characterHit); if (trailing != null) trailing[0] = OS.CharacterHit_TrailingLength(characterHit); OS.GCHandle_Free(characterHit); return untranslateOffset(offset); } /** * Returns the orientation of the receiver. * * @return the orientation style * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getOrientation () { checkLayout(); return orientation; } /** * Returns the previous offset for the specified offset and movement * type. The movement is one of <code>SWT.MOVEMENT_CHAR</code>, * <code>SWT.MOVEMENT_CLUSTER</code> or <code>SWT.MOVEMENT_WORD</code>, * <code>SWT.MOVEMENT_WORD_END</code> or <code>SWT.MOVEMENT_WORD_START</code>. * * @param offset the start offset * @param movement the movement type * @return the previous offset * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the offset is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getNextOffset(int, int) */ public int getPreviousOffset (int offset, int movement) { checkLayout(); return _getOffset (offset, movement, false); } /** * Gets the ranges of text that are associated with a <code>TextStyle</code>. * * @return the ranges, an array of offsets representing the start and end of each * text style. * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getStyles() * * @since 3.2 */ public int[] getRanges () { checkLayout(); int[] result = new int[styles.length * 2]; int count = 0; for (int i=0; i<styles.length - 1; i++) { if (styles[i].style != null) { result[count++] = styles[i].start; result[count++] = styles[i + 1].start - 1; } } if (count != result.length) { int[] newResult = new int[count]; System.arraycopy(result, 0, newResult, 0, count); result = newResult; } return result; } /** * Returns the text segments offsets of the receiver. * * @return the text segments offsets * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int[] getSegments () { checkLayout(); return segments; } /** * Returns the segments characters of the receiver. * * @return the segments characters * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @since 3.6 */ public char[] getSegmentsChars () { checkLayout(); return segmentsChars; } String getSegmentsText() { if (segments == null) return text; int nSegments = segments.length; if (nSegments <= 1) return text; int length = text.length(); if (length == 0) return text; if (nSegments == 2) { if (segments[0] == 0 && segments[1] == length) return text; } char[] oldChars = new char[length]; text.getChars(0, length, oldChars, 0); char[] newChars = new char[length + nSegments]; int charCount = 0, segmentCount = 0; char separator = orientation == SWT.RIGHT_TO_LEFT ? RTL_MARK : LTR_MARK; while (charCount < length) { if (segmentCount < nSegments && charCount == segments[segmentCount]) { newChars[charCount + segmentCount++] = separator; } else { newChars[charCount + segmentCount] = oldChars[charCount++]; } } if (segmentCount < nSegments) { segments[segmentCount] = charCount; newChars[charCount + segmentCount++] = separator; } return new String(newChars, 0, Math.min(charCount + segmentCount, newChars.length)); } /** * Returns the line spacing of the receiver. * * @return the line spacing * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getSpacing () { checkLayout(); return lineSpacing; } /** * Gets the style of the receiver at the specified character offset. * * @param offset the text offset * @return the style or <code>null</code> if not set * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the character offset is out of range</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public TextStyle getStyle (int offset) { checkLayout(); int length = text.length(); if (!(0 <= offset && offset < length)) SWT.error(SWT.ERROR_INVALID_RANGE); for (int i=1; i<styles.length; i++) { if (styles[i].start > offset) { return styles[i - 1].style; } } return null; } /** * Gets all styles of the receiver. * * @return the styles * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #getRanges() * * @since 3.2 */ public TextStyle[] getStyles () { checkLayout(); TextStyle[] result = new TextStyle[styles.length]; int count = 0; for (int i=0; i<styles.length; i++) { if (styles[i].style != null) { result[count++] = styles[i].style; } } if (count != result.length) { TextStyle[] newResult = new TextStyle[count]; System.arraycopy(result, 0, newResult, 0, count); result = newResult; } return result; } /** * Returns the tab list of the receiver. * * @return the tab list * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int[] getTabs () { checkLayout(); return tabs; } int GetTextRun(int textSourceCharacterIndex) { if (runs == null) runs = new int[4]; int index = 0; while (index < runs.length && runs[index] != 0) index++; if (index == runs.length) { int[] tmpRuns = new int[index + 4]; System.arraycopy(runs, 0, tmpRuns, 0, index); runs = tmpRuns; } int length = OS.String_Length(string); if (textSourceCharacterIndex >= length) { runs[index] = OS.gcnew_TextEndOfParagraph(1, defaultTextProperties); } else { int styleIndex = 1; while (styleIndex < styles.length && styles[styleIndex].start <= textSourceCharacterIndex) styleIndex++; TextStyle textStyle = styles[styleIndex - 1].style; int textProperties = styles[styleIndex - 1].textProperties; if (textProperties == 0) textProperties = defaultTextProperties; int end = styles[styleIndex].start; if (textStyle != null && textStyle.metrics != null) { GlyphMetrics metrics = textStyle.metrics; runs[index] = OS.gcnew_SWTTextEmbeddedObject(textProperties, end - textSourceCharacterIndex, metrics.width, metrics.ascent + metrics.descent, metrics.ascent); } else { char ch = segmentsText.charAt(textSourceCharacterIndex); if (ch == '\n' || ch == '\r') { int breakLength = 1; if (ch == '\r' && textSourceCharacterIndex + 1 < end && segmentsText.charAt(textSourceCharacterIndex + 1) == '\n') breakLength++; runs[index] = OS.gcnew_TextEndOfLine(breakLength, textProperties); } else { int i = textSourceCharacterIndex; while (i < end && (ch = segmentsText.charAt(i)) != '\n' && ch != '\r') i++; runs[index] = OS.gcnew_TextCharacters(string, textSourceCharacterIndex, i - textSourceCharacterIndex, textProperties); } } } return runs[index]; } int GetPrecedingText(int textSourceCharacterIndexLimit) { return 0; } /** * Gets the receiver's text, which will be an empty * string if it has never been set. * * @return the receiver's text * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public String getText () { checkLayout(); return text; } /** * Returns the width of the receiver. * * @return the width * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public int getWidth () { checkLayout(); return wrapWidth; } /** * Returns the receiver's wrap indent. * * @return the receiver's wrap indent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @since 3.6 */ public int getWrapIndent () { checkLayout(); return wrapIndent; } /** * Returns <code>true</code> if the text layout has been disposed, * and <code>false</code> otherwise. * <p> * This method gets the dispose state for the text layout. * When a text layout has been disposed, it is an error to * invoke any other method (except {@link #dispose()}) using the text layout. * </p> * * @return <code>true</code> when the text layout is disposed and <code>false</code> otherwise */ public boolean isDisposed () { return device == null; } /** * Sets the text alignment for the receiver. The alignment controls * how a line of text is positioned horizontally. The argument should * be one of <code>SWT.LEFT</code>, <code>SWT.RIGHT</code> or <code>SWT.CENTER</code>. * <p> * The default alignment is <code>SWT.LEFT</code>. Note that the receiver's * width must be set in order to use <code>SWT.RIGHT</code> or <code>SWT.CENTER</code> * alignment. * </p> * * @param alignment the new alignment * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setWidth(int) */ public void setAlignment (int alignment) { checkLayout(); int mask = SWT.LEFT | SWT.CENTER | SWT.RIGHT; alignment &= mask; if (alignment == 0) return; if ((alignment & SWT.LEFT) != 0) alignment = SWT.LEFT; if ((alignment & SWT.RIGHT) != 0) alignment = SWT.RIGHT; if (this.alignment == alignment) return; freeRuns(); this.alignment = alignment; } /** * Sets the ascent of the receiver. The ascent is distance in pixels * from the baseline to the top of the line and it is applied to all * lines. The default value is <code>-1</code> which means that the * ascent is calculated from the line fonts. * * @param ascent the new ascent * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the ascent is less than <code>-1</code></li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setDescent(int) * @see #getLineMetrics(int) */ public void setAscent(int ascent) { checkLayout(); if (ascent < -1) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (this.ascent == ascent) return; freeRuns(); this.ascent = ascent; } /** * Sets the descent of the receiver. The descent is distance in pixels * from the baseline to the bottom of the line and it is applied to all * lines. The default value is <code>-1</code> which means that the * descent is calculated from the line fonts. * * @param descent the new descent * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the descent is less than <code>-1</code></li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setAscent(int) * @see #getLineMetrics(int) */ public void setDescent(int descent) { checkLayout(); if (descent < -1) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (this.descent == descent) return; freeRuns(); this.descent = descent; } /** * Sets the default font which will be used by the receiver * to draw and measure text. If the * argument is null, then a default font appropriate * for the platform will be used instead. Note that a text * style can override the default font. * * @param font the new font for the receiver, or null to indicate a default font * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the font has been disposed</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setFont (Font font) { checkLayout(); if (font != null && font.isDisposed()) SWT.error(SWT.ERROR_INVALID_ARGUMENT); Font oldFont = this.font; if (oldFont == font) return; this.font = font; if (oldFont != null && oldFont.equals(font)) return; freeRuns(); } /** * Sets the indent of the receiver. This indent is applied to the first line of * each paragraph. * * @param indent new indent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setWrapIndent(int) * * @since 3.2 */ public void setIndent (int indent) { checkLayout(); if (indent < 0) return; if (this.indent == indent) return; freeRuns(); this.indent = indent; } /** * Sets the justification of the receiver. Note that the receiver's * width must be set in order to use justification. * * @param justify new justify * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @since 3.2 */ public void setJustify (boolean justify) { checkLayout(); if (this.justify == justify) return; freeRuns(); this.justify = justify; } /** * Sets the orientation of the receiver, which must be one * of <code>SWT.LEFT_TO_RIGHT</code> or <code>SWT.RIGHT_TO_LEFT</code>. * * @param orientation new orientation style * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setOrientation (int orientation) { checkLayout(); int mask = SWT.LEFT_TO_RIGHT | SWT.RIGHT_TO_LEFT; orientation &= mask; if (orientation == 0) return; if ((orientation & SWT.LEFT_TO_RIGHT) != 0) orientation = SWT.LEFT_TO_RIGHT; if (this.orientation == orientation) return; this.orientation = orientation; freeRuns(); } /** * Sets the offsets of the receiver's text segments. Text segments are used to * override the default behavior of the bidirectional algorithm. * Bidirectional reordering can happen within a text segment but not * between two adjacent segments. * <p> * Each text segment is determined by two consecutive offsets in the * <code>segments</code> arrays. The first element of the array should * always be zero and the last one should always be equals to length of * the text. * </p> * <p> * When segments characters are set, the segments are the offsets where * the characters are inserted in the text. * <p> * * @param segments the text segments offset * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setSegmentsChars(char[]) */ public void setSegments(int[] segments) { checkLayout(); if (this.segments == null && segments == null) return; if (this.segments != null && segments != null) { if (this.segments.length == segments.length) { int i; for (i = 0; i <segments.length; i++) { if (this.segments[i] != segments[i]) break; } if (i == segments.length) return; } } freeRuns(); this.segments = segments; } /** * Sets the characters to be used in the segments boundaries. The segments * are set by calling <code>setSegments(int[])</code>. The application can * use this API to insert Unicode Control Characters in the text to control * the display of the text and bidi reordering. The characters are not * accessible by any other API in <code>TextLayout</code>. * * @param segmentsChars the segments characters * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setSegments(int[]) * * @since 3.6 */ public void setSegmentsChars(char[] segmentsChars) { checkLayout(); if (this.segmentsChars == null && segmentsChars == null) return; if (this.segmentsChars != null && segmentsChars != null) { if (this.segmentsChars.length == segmentsChars.length) { int i; for (i = 0; i <segmentsChars.length; i++) { if (this.segmentsChars[i] != segmentsChars[i]) break; } if (i == segmentsChars.length) return; } } freeRuns(); this.segmentsChars = segmentsChars; } /** * Sets the line spacing of the receiver. The line spacing * is the space left between lines. * * @param spacing the new line spacing * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the spacing is negative</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setSpacing (int spacing) { checkLayout(); if (spacing < 0) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (this.lineSpacing == spacing) return; freeRuns(); this.lineSpacing = spacing; } /** * Sets the style of the receiver for the specified range. Styles previously * set for that range will be overwritten. The start and end offsets are * inclusive and will be clamped if out of range. * * @param style the style * @param start the start offset * @param end the end offset * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setStyle (TextStyle style, int start, int end) { checkLayout(); int length = text.length(); if (length == 0) return; if (start > end) return; start = Math.min(Math.max(0, start), length - 1); end = Math.min(Math.max(0, end), length - 1); int low = -1; int high = styles.length; while (high - low > 1) { int index = (high + low) / 2; if (styles[index + 1].start > start) { high = index; } else { low = index; } } if (0 <= high && high < styles.length) { StyleItem item = styles[high]; if (item.start == start && styles[high + 1].start - 1 == end) { if (style == null) { if (item.style == null) return; } else { if (style.equals(item.style)) return; } } } freeRuns(); int modifyStart = high; int modifyEnd = modifyStart; while (modifyEnd < styles.length) { if (styles[modifyEnd + 1].start > end) break; modifyEnd++; } if (modifyStart == modifyEnd) { int styleStart = styles[modifyStart].start; int styleEnd = styles[modifyEnd + 1].start - 1; if (styleStart == start && styleEnd == end) { styles[modifyStart].style = style; return; } if (styleStart != start && styleEnd != end) { StyleItem[] newStyles = new StyleItem[styles.length + 2]; System.arraycopy(styles, 0, newStyles, 0, modifyStart + 1); StyleItem item = new StyleItem(); item.start = start; item.style = style; newStyles[modifyStart + 1] = item; item = new StyleItem(); item.start = end + 1; item.style = styles[modifyStart].style; newStyles[modifyStart + 2] = item; System.arraycopy(styles, modifyEnd + 1, newStyles, modifyEnd + 3, styles.length - modifyEnd - 1); styles = newStyles; return; } } if (start == styles[modifyStart].start) modifyStart--; if (end == styles[modifyEnd + 1].start - 1) modifyEnd++; int newLength = styles.length + 1 - (modifyEnd - modifyStart - 1); StyleItem[] newStyles = new StyleItem[newLength]; System.arraycopy(styles, 0, newStyles, 0, modifyStart + 1); StyleItem item = new StyleItem(); item.start = start; item.style = style; newStyles[modifyStart + 1] = item; styles[modifyEnd].start = end + 1; System.arraycopy(styles, modifyEnd, newStyles, modifyStart + 2, styles.length - modifyEnd); styles = newStyles; } /** * Sets the receiver's tab list. Each value in the tab list specifies * the space in pixels from the origin of the text layout to the respective * tab stop. The last tab stop width is repeated continuously. * * @param tabs the new tab list * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setTabs (int[] tabs) { checkLayout(); if (this.tabs == null && tabs == null) return; if (this.tabs != null && tabs !=null) { if (this.tabs.length == tabs.length) { int i; for (i = 0; i <tabs.length; i++) { if (this.tabs[i] != tabs[i]) break; } if (i == tabs.length) return; } } freeRuns(); this.tabs = tabs; } /** * Sets the receiver's text. *<p> * Note: Setting the text also clears all the styles. This method * returns without doing anything if the new text is the same as * the current text. * </p> * * @param text the new text * * @exception IllegalArgumentException <ul> * <li>ERROR_NULL_ARGUMENT - if the text is null</li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> */ public void setText (String text) { checkLayout(); if (text == null) SWT.error(SWT.ERROR_NULL_ARGUMENT); if (text.equals(this.text)) return; freeRuns(); this.text = text; styles = new StyleItem[2]; styles[0] = new StyleItem(); styles[1] = new StyleItem(); styles[1].start = text.length(); } /** * Sets the line width of the receiver, which determines how * text should be wrapped and aligned. The default value is * <code>-1</code> which means wrapping is disabled. * * @param width the new width * * @exception IllegalArgumentException <ul> * <li>ERROR_INVALID_ARGUMENT - if the width is <code>0</code> or less than <code>-1</code></li> * </ul> * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setAlignment(int) */ public void setWidth (int width) { checkLayout(); if (width < -1 || width == 0) SWT.error(SWT.ERROR_INVALID_ARGUMENT); if (this.wrapWidth == width) return; freeRuns(); this.wrapWidth = width; } /** * Sets the wrap indent of the receiver. This indent is applied to all lines * in the paragraph except the first line. * * @param wrapIndent new wrap indent * * @exception SWTException <ul> * <li>ERROR_GRAPHIC_DISPOSED - if the receiver has been disposed</li> * </ul> * * @see #setIndent(int) * * @since 3.6 */ public void setWrapIndent (int wrapIndent) { checkLayout(); if (wrapIndent < 0) return; if (this.wrapIndent == wrapIndent) return; freeRuns(); this.wrapIndent = wrapIndent; } int validadeOffset(int offset, int step) { offset += step; if (segments != null && segments.length > 2) { for (int i = 0; i < segments.length; i++) { if (translateOffset(segments[i]) - 1 == offset) { offset += step; break; } } } return offset; } /** * Returns a string containing a concise, human-readable * description of the receiver. * * @return a string representation of the receiver */ public String toString () { if (isDisposed()) return "TextLayout {*DISPOSED*}"; return "TextLayout {}"; } int translateOffset(int offset) { if (segments == null) return offset; int nSegments = segments.length; if (nSegments <= 1) return offset; int length = text.length(); if (length == 0) return offset; if (nSegments == 2) { if (segments[0] == 0 && segments[1] == length) return offset; } for (int i = 0; i < nSegments && offset - i >= segments[i]; i++) { offset++; } return offset; } int untranslateOffset(int offset) { if (segments == null) return offset; int nSegments = segments.length; if (nSegments <= 1) return offset; int length = text.length(); if (length == 0) return offset; if (nSegments == 2) { if (segments[0] == 0 && segments[1] == length) return offset; } for (int i = 0; i < nSegments && offset > segments[i]; i++) { offset--; } return offset; } }